Spring Boot 无侵入式 实现API接口统一JSON格式返回 您所在的位置:网站首页 spring boot 返回json Spring Boot 无侵入式 实现API接口统一JSON格式返回

Spring Boot 无侵入式 实现API接口统一JSON格式返回

#Spring Boot 无侵入式 实现API接口统一JSON格式返回| 来源: 网络整理| 查看: 265

 无侵入式 统一返回JSON格式

其实本没有没打算写这篇博客的,但还是要写一下写这篇博客的起因是因为,现在呆着的这家公司居然没有统一的API返回格式?,询问主管他居然告诉我用HTTP状态码就够用了(fxxk),天哪HTTP状态码真的够用吗?

在仔细的阅读了项目源码后发现,在API请求的是居然没有业务异常(黑人问好)。好吧 居然入坑了只能遵照项目风格了,懒得吐槽了。

因为项目已经开发了半年多了, 要是全部接口都做修改工作量还是挺大的, 只能用这种无侵入式的方案来解决.

项目源代码: github.com/469753862/g…

定义JSON格式

定义返回JSON格式

后端返回给前端一般情况下使用JSON格式, 定义如下

{     "code": 200,     "message": "OK",     "data": {     } } 复制代码

code: 返回状态码 message: 返回信息的描述 data: 返回值 定义JavaBean字段

定义状态码枚举类

@ToString @Getter public enum ResultStatus {     SUCCESS(HttpStatus.OK, 200, "OK"),     BAD_REQUEST(HttpStatus.BAD_REQUEST, 400, "Bad Request"),     INTERNAL_SERVER_ERROR(HttpStatus.INTERNAL_SERVER_ERROR, 500, "Internal Server Error"),;     /** 返回的HTTP状态码,  符合http请求 */     private HttpStatus httpStatus;     /** 业务异常码 */     private Integer code;     /** 业务异常信息描述 */     private String message;     ResultStatus(HttpStatus httpStatus, Integer code, String message) {         this.httpStatus = httpStatus;         this.code = code;         this.message = message;     } } 复制代码

状态码和信息以及http状态码就能一一对应了便于维护, 有同学有疑问了为什么要用到http状态码呀,因为我要兼容项目以前的代码, 没有其他原因, 当然其他同学不喜欢http状态码的可以吧源码中HttpStatus给删除了

定义返回体类

@Getter @ToString public class Result {     /** 业务错误码 */     private Integer code;     /** 信息描述 */     private String message;     /** 返回参数 */     private T data;     private Result(ResultStatus resultStatus, T data) {         this.code = resultStatus.getCode();         this.message = resultStatus.getMessage();         this.data = data;     }     /** 业务成功返回业务代码和描述信息 */     public static Result success() {         return new Result(ResultStatus.SUCCESS, null);     }     /** 业务成功返回业务代码,描述和返回的参数 */     public static  Result success(T data) {         return new Result(ResultStatus.SUCCESS, data);     }     /** 业务成功返回业务代码,描述和返回的参数 */     public static  Result success(ResultStatus resultStatus, T data) {         if (resultStatus == null) {             return success(data);         }         return new Result(resultStatus, data);     }     /** 业务异常返回业务代码和描述信息 */     public static  Result failure() {         return new Result(ResultStatus.INTERNAL_SERVER_ERROR, null);     }     /** 业务异常返回业务代码,描述和返回的参数 */     public static  Result failure(ResultStatus resultStatus) {         return failure(resultStatus, null);     }     /** 业务异常返回业务代码,描述和返回的参数 */     public static  Result failure(ResultStatus resultStatus, T data) {         if (resultStatus == null) {             return new Result(ResultStatus.INTERNAL_SERVER_ERROR, null);         }         return new Result(resultStatus, data);     } } 复制代码

因为使用构造方法进行创建对象太麻烦了, 我们使用静态方法来创建对象这样简单明了

Result实体返回测试 @RestController @RequestMapping("/hello") public class HelloController {     private static final HashMap INFO;     static {         INFO = new HashMap();         INFO.put("name", "galaxy");         INFO.put("age", "70");     }     @GetMapping("/hello")     public Map hello() {         return INFO;     }     @GetMapping("/result")     @ResponseBody     public Result helloResult() {         return Result.success(INFO);     } } 复制代码

到这里我们已经简单的实现了统一JSON格式了, 但是我们也发现了一个问题了,想要返回统一的JSON格式需要返回Result才可以, 我明明返回Object可以了, 为什么要重复劳动, 有没有解决方法, 当然是有的啦, 下面我们开始优化我们的代码吧

统一返回JSON格式进阶-全局处理(@RestControllerAdvice)

我师傅经常告诉我的一句话: “你就是一个小屁孩, 你遇到的问题都已经不知道有多少人遇到过了, 你会想到的问题, 已经有前辈想到过了. 你准备解决的问题, 已经有人把坑填了”。是不是很鸡汤, 是不是很励志, 让我对前辈们充满着崇拜, 事实上他对我说的是: “自己去百度”, 这五个大字, 其实这五个大字已经说明上明的B话了, 通过不断的百度和Google发现了很多的解决方案.

我们都知道使用@ResponseBody注解会把返回Object序列化成JSON字符串,就先从这个入手吧, 大致就是在序列化前把Object赋值给Result就可以了, 大家可以观摩org.springframework.web.servlet.mvc.method.annotation.ResponseBodyAdvice和org.springframework.web.bind.annotation.ResponseBody

@ResponseBody继承类

我们已经决定从@ResponseBody注解入手了就创建一个注解类继承@ResponseBody, 很干净什么都没有哈哈,@ResponseResultBody 可以标记在类和方法上这样我们就可以跟自由的进行使用了

@Retention(RetentionPolicy.RUNTIME) @Target({ElementType.TYPE, ElementType.METHOD}) @Documented @ResponseBody public @interface ResponseResultBody { } 复制代码

ResponseBodyAdvice继承类 @RestControllerAdvice public class ResponseResultBodyAdvice implements ResponseBodyAdvice {     private static final Class> converterType) {         return AnnotatedElementUtils.hasAnnotation(returnType.getContainingClass(), ANNOTATION_TYPE) || returnType.hasMethodAnnotation(ANNOTATION_TYPE);     }     /** 当类或者方法使用了 @ResponseResultBody 就会调用这个方法 */     @Override     public Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType, Class> selectedConverterType, ServerHttpRequest request, ServerHttpResponse response) {         if (body instanceof Result) {             return body;         }         return Result.success(body);     }     /**      * 提供对标准Spring MVC异常的处理      *      * @param ex      the target exception      * @param request the current request      */     @ExceptionHandler(Exception.class)     public final ResponseEntity> handleResultException(ResultException ex, HttpHeaders headers, WebRequest request) {         Result body = Result.failure(ex.getResultStatus());         HttpStatus status = ex.getResultStatus().getHttpStatus();         return this.handleExceptionInternal(ex, body, headers, status, request);     }     /** 异常类的统一处理 */     protected ResponseEntity body = Result.failure();         HttpStatus status = HttpStatus.INTERNAL_SERVER_ERROR;         return this.handleExceptionInternal(ex, body, headers, status, request);     }     /**      * org.springframework.web.servlet.mvc.method.annotation.ResponseEntityExceptionHandler#handleExceptionInternal(java.lang.Exception, java.lang.Object, org.springframework.http.HttpHeaders, org.springframework.http.HttpStatus, org.springframework.web.context.request.WebRequest)      * 

     * A single place to customize the response body of all exception types.      * 

The default implementation sets the {@link WebUtils#ERROR_EXCEPTION_ATTRIBUTE}      * request attribute and creates a {@link ResponseEntity} from the given      * body, headers, and status.      */     protected ResponseEntity



【本文地址】

公司简介

联系我们

今日新闻

    推荐新闻

    专题文章
      CopyRight 2018-2019 实验室设备网 版权所有